Environment Management
IMO, Keeping all language versions in a dedicated ~/language-versions
folder is the cleanest and most explicit way to manage multiple versions without cluttering system paths. It gives you:
✅ Full control over versions (install, remove, switch at will).
✅ No clutter in /usr/local/bin
, /usr/bin
, or $HOME/bin
.
✅ No conflicts with system-wide installations.
✅ Explicit switching via PATH
, aliases, or symlinks.
🚀 A Universal Multi-Language Versioning Setup
If you want to apply this approach to Go, Python, Node.js, Rust, etc., you can structure it like this:
~/language-versions/
├── go/
│ ├── go1.22.9/
│ ├── go1.21.6/
│ ├── go1.18.10/
│ ├── current/ # (symlink to active version)
│
├── python/
│ ├── python3.12/
│ ├── python3.11/
│ ├── python3.10/
│ ├── current/ # (symlink to active version)
│
├── node/
│ ├── node18/
│ ├── node16/
│ ├── node14/
│ ├── current/ # (symlink to active version)
1️⃣ Installing Languages in ~/language-versions
🔹 Go (Explicit Versioning)
mkdir -p ~/language-versions/go
cd ~/language-versions/go
# Download and extract Go versions
curl -LO https://go.dev/dl/go1.22.9.linux-amd64.tar.gz
tar -xzf go1.22.9.linux-amd64.tar.gz
mv go go1.22.9
Do the same for go1.21.6
, go1.18.10
, etc.
🔹 Python (Manual Install)
mkdir -p ~/language-versions/python
cd ~/language-versions/python
# Install Python versions
sudo apt install python3.12 python3.11 python3.10 # Ubuntu
If using source:
curl -LO https://www.python.org/ftp/python/3.12.0/Python-3.12.0.tgz
tar -xzf Python-3.12.0.tgz
mv Python-3.12.0 python3.12
🔹 Node.js (Manual or n
Tool)
mkdir -p ~/language-versions/node
cd ~/language-versions/node
# Install Node.js versions manually
curl -LO https://nodejs.org/dist/v18.17.0/node-v18.17.0-linux-x64.tar.gz
tar -xzf node-v18.17.0-linux-x64.tar.gz
mv node-v18.17.0-linux-x64 node18
2️⃣ Switching Versions Explicitly
You can switch versions explicitly by modifying PATH
:
Go Example
export PATH=~/language-versions/go/go1.22.9/bin:$PATH
go version # Shows Go 1.22.9
Python Example
export PATH=~/language-versions/python/python3.12/bin:$PATH
python --version # Shows Python 3.12
Node.js Example
export PATH=~/language-versions/node/node18/bin:$PATH
node -v # Shows Node.js v18
3️⃣ Use Symlinks for Easier Switching
Instead of manually changing PATH
, create a current
symlink:
cd ~/language-versions/go
ln -sfn go1.22.9 current # Switch Go version
Then, just add one-time setup to .bashrc
or .zshrc
:
export PATH=~/language-versions/go/current/bin:$PATH
Now, every time you change current
, it instantly switches to the new version.
Same idea for Python:
cd ~/language-versions/python
ln -sfn python3.12 current # Switch Python version
And Node.js:
cd ~/language-versions/node
ln -sfn node18 current # Switch Node.js version
Now you can switch versions just by updating the symlink!
4️⃣ (Optional) Create Aliases for Quick Switching
Add these to your ~/.bashrc
or ~/.zshrc
:
alias use-go1.22="ln -sfn ~/language-versions/go/go1.22.9 ~/language-versions/go/current"
alias use-go1.21="ln -sfn ~/language-versions/go/go1.21.6 ~/language-versions/go/current"
alias use-python3.12="ln -sfn ~/language-versions/python/python3.12 ~/language-versions/python/current"
alias use-node18="ln -sfn ~/language-versions/node/node18 ~/language-versions/node/current"
Now switching is explicit and quick:
use-go1.22
go version # Now using Go 1.22.9
🚀 Why This Method Rocks
✔ No system-wide clutter (/usr/bin
, /usr/local/bin
stay clean).
✔ Explicit version control (no magic happening behind the scenes).
✔ No conflicts (system Go/Python/Node remain untouched).
✔ Easy switching with ln -sfn
or export PATH
.
✔ Works across all languages (not tied to a tool like nvm
, pyenv
, or gvm
).
🎯 Final Thoughts
This method is basically "nvm/pyenv/gvm but explicit and self-contained."
- Works on any machine, with any language.
- No system-wide modifications.
- You always know exactly where everything lives.
Additionally, I maintain a structured environment setup:
~/env_setup/<project>/local_backend.sh # or 'prod', 'staging', etc.
I always source the appropriate file when opening a new shell, ensuring environment consistency without persisting changes beyond the current session:
source ~/env_setup/my_project/local_backend.sh